home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1999 March - Disc 1 / Macworld (1999-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / SystemCode / appleEvents.tcl < prev    next >
Encoding:
Text File  |  1998-11-21  |  9.2 KB  |  322 lines  |  [TEXT/ALFA]

  1. # JEG - modernized
  2. # make alias list to pass to AEBuild
  3. proc makeAlis {name} {
  4.     return [aebuild::alis $name]
  5. }
  6.  
  7. # JEG - This is unused???
  8. proc makeFile {name} {
  9.     return [aebuild::alis $name]    
  10. }
  11.  
  12. ## 
  13.  # -------------------------------------------------------------------------
  14.  # 
  15.  # "makeAlises" --
  16.  # 
  17.  #  This proc has changed so it takes a list of items rather than an
  18.  #  unknown number of args 'args'.  If 'l' is a list you must call
  19.  #  this proc with 'makeAlises $l' rather than 'eval makeAlises $l'
  20.  #  as was previously required.
  21.  # -------------------------------------------------------------------------
  22.  ##
  23.  
  24. # JEG - modernized
  25. proc makeAlises {names} {
  26.     return [aebuild::list $names -as alis]
  27. }
  28.  
  29. ## 
  30.  # -------------------------------------------------------------------------
  31.  # 
  32.  # "handleReply" --
  33.  # 
  34.  #  Queued replies are passed through AEPrint and then to this routine.
  35.  #  
  36.  #  If you write your own handleReply procedure, register it to this
  37.  #  proc with:
  38.  #  
  39.  #    currentReplyHandler 'my-proc-name'
  40.  #    
  41.  #  Do this each time you send an event which may receive a reply.
  42.  #  There is no need to register your proc at startup or any such
  43.  #  'pre-registering'.  Just call the above proc _each_ time.
  44.  #  
  45.  #  You proc should take one parameter (the reply), and should
  46.  #  return '1' if it handled the reply, otherwise it can do/return
  47.  #  anything else (although hopefully not much if it didn't handle
  48.  #  anything).
  49.  #  
  50.  #  If your replies often time-out or have other problems such
  51.  #  that you don't handle them correctly, you may wish to register
  52.  #  your reply-handler with 'currentReplyHandler 'my-proc' 1' which
  53.  #  says 'only register if it's not already registered'.  Or you
  54.  #  may wish to remove duplicates from the list of handlers 
  55.  #  directly.
  56.  #    
  57.  # Results:
  58.  #  depends on what is registered
  59.  # 
  60.  # Side effects:
  61.  #  calls other procs.  Removes handler from queue if it handled
  62.  #  the reply.
  63.  # 
  64.  # --Version--Author------------------Changes-------------------------------
  65.  #    1.0     <darley@fas.harvard.edu> first one with hook handling
  66.  #    2.0     <darley@fas.harvard.edu> different mechanism to give priority
  67.  # -------------------------------------------------------------------------
  68.  ##
  69. proc handleReply {rep} {
  70.     global lastReply replyHandlers
  71.     set lastReply $rep
  72.     set i 0
  73.     foreach h $replyHandlers {
  74.     if {$h != ""} {
  75.         catch [list $h $rep] res
  76.         if {$res == 1} {
  77.         set replyHandlers [lreplace $replyHandlers $i $i]
  78.         return
  79.         }
  80.     }
  81.     incr i
  82.     }
  83.     message "Reply '$rep' not handled"
  84. }
  85.  
  86. ensureset replyHandlers ""
  87.  
  88. ## 
  89.  # -------------------------------------------------------------------------
  90.  # 
  91.  # "currentReplyHandler" --
  92.  # 
  93.  #  Add item to end of queue to receive replies, even if it is already
  94.  #  in the queue, unless we set 'nodups'
  95.  # -------------------------------------------------------------------------
  96.  ##
  97. proc currentReplyHandler {proc {nodups 0}} {
  98.     global replyHandlers
  99.     if {!$nodups || (![lcontains replyHandlers $proc])} {
  100.     lappend replyHandlers $proc
  101.     }    
  102. }
  103.  
  104. # JEG - only used by thinkMenu.tcl. Why is it here?
  105. # Return an object record specifying the desired think project file.
  106. proc fileObject {name} {
  107.     join [concat {obj\{want:type('SFIL'), from:'null'(), form:'name', seld:“} [file tail $name] {”\}}] ""
  108. }
  109.  
  110. proc sendOpenEvent {filler appname fname} {
  111.     if {$filler == "noReply"} {
  112.     AEBuild $appname aevt odoc "----" [makeAlis $fname]
  113.     } else {
  114.     AEBuild -r $appname aevt odoc "----" [makeAlis $fname]
  115.     }
  116. }
  117.  
  118.  
  119. # Send open folder event to Finder. Name must end in colon.
  120. proc openFolder {name} {
  121.     if {![regexp ":$" $name]} {
  122.     append name ":"
  123.     }
  124.     switchTo Finder
  125.     sendOpenEvent -r Finder $name
  126. }
  127.  
  128. proc launchDoc {name} {
  129.     set app [app::launchFore [getFileSig $name]]
  130.     sendOpenEvent -r [file tail $app] $name
  131. }
  132.  
  133. # Send multiple open events
  134. proc sendOpenEvents {appname args} {
  135.     AEBuild -r $appname aevt odoc "----" [makeAlises $args]
  136. }
  137.  
  138. proc openAndSendFile {sig} {
  139.     set fname [win::Current]
  140.     if {[winDirty]} {
  141.     if {[dialog::yesno "Save '$fname'?"]} {
  142.         save
  143.     }
  144.     }
  145.     
  146.     set name [file tail [app::launchFore $sig]]
  147.     sendOpenEvent noReply $name $fname
  148. }
  149.  
  150. #================================================================================
  151. # General Apple Event handling routines
  152. #
  153. # (written by Tom Pollard for use in the MacPerl package)
  154. #================================================================================
  155.  
  156. # Quit an application.
  157. proc sendQuitEvent {appname} {
  158.     AEBuild $appname "aevt" "quit" 
  159. }
  160.  
  161. # Close one of an application's windows, designated by number.
  162. proc sendCloseWinNum {appname num} {
  163.     AEBuild $appname "core" "clos" "----" [AEWinByPos $num]
  164. }
  165.  
  166. # Close one of an application's windows, designated by name.
  167. proc sendCloseWinName {appname name} {
  168.     AEBuild $appname "core" "clos" "----" [AEWinByName $name]
  169. }
  170.  
  171. # Obtain the number of lines in one of an application's
  172. # windows, designated by name.
  173. proc sendCountLines {appname name} {
  174.     set winObj [AEWinByName $name]
  175.     set res [AEBuild -r $appname "core" "cnte" "----" $winObj kocl type('clin')]    
  176.     if {[regexp {:(.*)\}} $res allofit nlines]} {
  177.     return $nlines
  178.     } else {
  179.     return 0
  180.     }
  181. }
  182.  
  183. # Get a selected range of lines from one of an application's
  184. # windows, designated by name.  If $last is missing, then a single
  185. # line is returned; if both $first and $last are missing, then
  186. # the complete window contents are returned.
  187. proc sendGetText {appname name {first {missing}} {last {missing}}} {
  188.     global ALPHA
  189.     set winObj [AEWinByName $name]
  190.     if {$first != "missing"} {
  191.     if {$last != "missing"} {
  192.         set rangDesc [AELineRange $first $last]
  193.     } else {
  194.         set rangDesc [AEAbsPos $first]
  195.     }
  196.     set objDesc "obj{want:type('clin'), from:$winObj, $rangDesc }"
  197.     } else {
  198.     set objDesc "obj{want:type('ctxt'), from:$winObj, form:'indx', seld:abso('all') }"
  199.     }
  200.     set res [AEBuild -r $appname "core" "getd" "----" $objDesc]
  201.     if {![regexp {“.*”} $res text]} { set text {} }
  202.     return [string trim $text {“”}]
  203. }
  204.  
  205. # Set a selected range of lines in one of an application's
  206. # windows, designated by name.  If $last is missing, then a single
  207. # line is changed; if both $first and $last are missing, then
  208. # the complete window contents are replaced by the new text.
  209. proc sendSetText {appname name text {first {missing}} {last {missing}}} {
  210.     set winObj [AEWinByName $name]
  211.     if {$first != "missing"} {
  212.     if {$last != "missing"} {
  213.         set rangDesc [AELineRange $first $last]
  214.     } else {
  215.         set rangDesc [AEAbsPos $first]
  216.     }
  217.     set objDesc "obj{want:type('clin'), from:$winObj, $rangDesc }"
  218.     } else {
  219.     set objDesc "obj{want:type('ctxt'), from:$winObj, form:'indx', seld:abso('all') }"
  220.     }
  221.     set res [AEBuild -r $appname "core" "setd" "----" $objDesc "data" [curlyq $text]]    
  222.     if {![regexp {“.*”} $res text]} { set text {} }
  223.     return [string trim $text {“”}]
  224. }
  225.  
  226. ################################################################################
  227. # Utility functions for constructing AppleEvent descriptors for AEBuild
  228. ################################################################################
  229.  
  230. # JEG - modernized
  231. #
  232. proc AEFilename {name} {
  233.     return [aebuild::filename $name]
  234. }
  235.  
  236. # JEG - modernized
  237. #
  238. proc AEWinByName {name} {
  239.     return [aebuild::winByName $name]
  240. }
  241.  
  242. # JEG - modernized
  243. #
  244. proc AEWinByPos {absPos} {
  245.     return [aebuild::winByPos $absPos]
  246. }
  247.  
  248. # JEG - modernized
  249. #
  250. proc AELineRange {absPos1 absPos2} {
  251.     return [aebuild::lineRange $absPos1 $absPos2]
  252. }
  253.  
  254. # JEG - modernized
  255. #
  256. proc AEAbsPos {posName} {
  257.     return [aebuild::absPos $posName]
  258. }
  259.  
  260. # JEG - modernized
  261. #
  262. proc AEName {name} {
  263.     return [aebuild::name $name]
  264. }
  265.  
  266. # JEG - modernized
  267. #
  268. proc curlyq {str} {
  269.     return [aebuild::TEXT $str]
  270. }
  271.  
  272. ################################################################################
  273. proc nullObject {}                     { return "'null'()" }
  274. proc objectType {type}                 { return "type($type)" }
  275. proc nameObject {type name from}     { return "obj \{form:name, want:[objectType $type], seld:$name, from:$from\}" }
  276. proc indexObject {type ind from}     { return "obj \{form:indx, want:[objectType $type], seld:$ind, from:$from\}" }
  277. proc propertyObject { prop object } { return "obj \{form:prop, want:[objectType prop], seld:[objectType $prop], from:$object\}" }
  278.  
  279. # JEG - unused?
  280. # 'process' must have single quotes
  281. proc buildMsgReply { process suite event args } { return [eval [list AEBuild -r $process $suite $event ] $args] }
  282.  
  283. # JEG - modernized
  284. #
  285. proc countObjects { process fromObject class } {
  286.     return [aebuild::result $process core cnte \
  287.       ---- $fromObject \
  288.       kocl [objectType $class] \
  289.       ]
  290. }
  291.  
  292. proc createThingAtEnd {process container class} {
  293.     set res [AEBuild -r $process core crel insh "insl \{kobj:$container\}" kocl "type($class)"]
  294. }
  295.  
  296.  
  297. proc getObjectData { process class name from } {
  298.     set res [AEBuild -r $process core getd ---- [nameObject $class "“$name”" $from] {rtyp{type:TEXT}}]
  299.     if {[regexp {“(.*)”} $res dummy mtch]} {
  300.     return $mtch
  301.     } else {
  302.     error "Bad count proc"
  303.     }
  304. }
  305.  
  306.  
  307. proc objectProperty { process property object } {
  308.     AEBuild -r $process core getd ---- [propertyObject $property $object]
  309. }
  310.  
  311. # Extract and return a path from a result.
  312. proc extractPath {res} {
  313.     if {[regexp {«(.*)»} $res dummy fss]} {
  314.     return [specToPathName $fss]
  315.     }
  316.     error "bad path $name"
  317. }    
  318.